学习多线程有一段时间了,一直没有实践,从网上找了一道经典题来练练手。
题目
启动3个线程打印递增的数字,线程1先打印1,2,3,4,5,然后是线程2打印6,7,8,9,10,然后是线程3打印11,12,13,14 15。接着再由线程1打印16,17,18,19,20 ….以此类推,直到打印到75。程序的输出结果应该为:
线程1:1
线程1:2
线程1:3
线程1:4
线程1:5
线程2:6
线程2:7
线程2:8
线程2:9
线程2:10
……
解答一
该解法忽略了 synchronized 隐式锁只有一个任务等待队列,不能指定唤醒哪一个线程,因此会有不定的等待时间。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83
| package printnum;
import java.text.SimpleDateFormat; import java.util.Date;
public class PrintNum { private static final short MAX_NUM = 75; private static final SimpleDateFormat DF = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); private volatile short num = 0; private volatile boolean firstCondiction = true; private volatile boolean secondCondiction = false; private volatile boolean thirdCondiction = false;
private synchronized void printNum(short threadIndex) throws InterruptedException { while (num <= MAX_NUM) { if ((firstCondiction && threadIndex == 1) || (secondCondiction && threadIndex == 2) || (thirdCondiction && threadIndex == 3)) { for (int i = 1; i <= 5; i++) { num++; if (num <= MAX_NUM) { System.out.println(DF.format(new Date()) + " 线程" + threadIndex + ":" + num); } } switch (threadIndex) { case 1: firstCondiction = false; secondCondiction = true; thirdCondiction = false; break; case 2: firstCondiction = false; secondCondiction = false; thirdCondiction = true; break; case 3: firstCondiction = true; thirdCondiction = false; secondCondiction = false; break; default: System.out.println("errorIndex:" + threadIndex); break; } } notifyAll(); wait(100); } wait(100); }
private void printAllNum() { for (short i = 1; i <= 3; i++) { final short threadIndex = i; Thread thread = new Thread(new Runnable() { public void run() { try { printNum(threadIndex); } catch (InterruptedException e) { e.printStackTrace(); } } }, "线程" + i); thread.start(); } }
public static void main(String[] args) { PrintNum printNum = new PrintNum(); printNum.printAllNum(); } }
|
运行结果如下(注意数字20和21之间等待了42s):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
| 2019-09-22 17:36:19 线程1:1 2019-09-22 17:36:19 线程1:2 2019-09-22 17:36:19 线程1:3 2019-09-22 17:36:19 线程1:4 2019-09-22 17:36:19 线程1:5 2019-09-22 17:36:19 线程2:6 2019-09-22 17:36:19 线程2:7 2019-09-22 17:36:19 线程2:8 2019-09-22 17:36:19 线程2:9 2019-09-22 17:36:19 线程2:10 2019-09-22 17:36:19 线程3:11 2019-09-22 17:36:19 线程3:12 2019-09-22 17:36:19 线程3:13 2019-09-22 17:36:19 线程3:14 2019-09-22 17:36:19 线程3:15 2019-09-22 17:36:19 线程1:16 2019-09-22 17:36:19 线程1:17 2019-09-22 17:36:19 线程1:18 2019-09-22 17:36:19 线程1:19 2019-09-22 17:36:19 线程1:20 2019-09-22 17:37:01 线程2:21 2019-09-22 17:37:01 线程2:22 2019-09-22 17:37:01 线程2:23 2019-09-22 17:37:01 线程2:24 2019-09-22 17:37:01 线程2:25 2019-09-22 17:37:01 线程3:26 2019-09-22 17:37:01 线程3:27 2019-09-22 17:37:01 线程3:28 2019-09-22 17:37:01 线程3:29 2019-09-22 17:37:01 线程3:30 2019-09-22 17:37:01 线程1:31 2019-09-22 17:37:01 线程1:32 2019-09-22 17:37:01 线程1:33 2019-09-22 17:37:01 线程1:34 2019-09-22 17:37:01 线程1:35 2019-09-22 17:37:01 线程2:36 2019-09-22 17:37:01 线程2:37 2019-09-22 17:37:01 线程2:38 2019-09-22 17:37:01 线程2:39 2019-09-22 17:37:01 线程2:40 2019-09-22 17:37:01 线程3:41 2019-09-22 17:37:01 线程3:42 2019-09-22 17:37:01 线程3:43 2019-09-22 17:37:01 线程3:44 2019-09-22 17:37:01 线程3:45 2019-09-22 17:37:01 线程1:46 2019-09-22 17:37:01 线程1:47 2019-09-22 17:37:01 线程1:48 2019-09-22 17:37:01 线程1:49 2019-09-22 17:37:01 线程1:50 2019-09-22 17:37:01 线程2:51 2019-09-22 17:37:01 线程2:52 2019-09-22 17:37:01 线程2:53 2019-09-22 17:37:01 线程2:54 2019-09-22 17:37:01 线程2:55 2019-09-22 17:37:01 线程3:56 2019-09-22 17:37:01 线程3:57 2019-09-22 17:37:01 线程3:58 2019-09-22 17:37:01 线程3:59 2019-09-22 17:37:01 线程3:60 2019-09-22 17:37:01 线程1:61 2019-09-22 17:37:01 线程1:62 2019-09-22 17:37:01 线程1:63 2019-09-22 17:37:01 线程1:64 2019-09-22 17:37:01 线程1:65 2019-09-22 17:37:01 线程2:66 2019-09-22 17:37:01 线程2:67 2019-09-22 17:37:01 线程2:68 2019-09-22 17:37:01 线程2:69 2019-09-22 17:37:01 线程2:70 2019-09-22 17:37:01 线程3:71 2019-09-22 17:37:01 线程3:72 2019-09-22 17:37:01 线程3:73 2019-09-22 17:37:01 线程3:74 2019-09-22 17:37:01 线程3:75
|
解答二
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
| package printnum;
import java.text.SimpleDateFormat; import java.util.Date; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.locks.Condition; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock;
public class ConditionPrintNum { private static final short MAX_NUM = 75; private static final SimpleDateFormat DF = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); private final Lock lock = new ReentrantLock(); private final Condition condition1 = lock.newCondition(); private final Condition condition2 = lock.newCondition(); private final Condition condition3 = lock.newCondition(); private AtomicInteger num = new AtomicInteger(0); private volatile short lastThreadIndex = 3;
private void printNum(Condition currentCondition, Condition signalCondition, short threadIndex) throws InterruptedException { while (num.intValue() <= MAX_NUM) { while ((lastThreadIndex == 3 && threadIndex != 1) || (lastThreadIndex == 1 && threadIndex != 2) || (lastThreadIndex == 2 && threadIndex != 3)) { signalCondition.signalAll(); currentCondition.await(100, TimeUnit.MILLISECONDS); } for (int i = 1; i <= 5; i++) { num.getAndAdd(1); if (num.intValue() <= MAX_NUM) { System.out.println(DF.format(new Date()) + " 线程" + threadIndex + ":" + num); } } lastThreadIndex = threadIndex; signalCondition.signalAll(); currentCondition.await(100, TimeUnit.MILLISECONDS); } currentCondition.await(100, TimeUnit.MILLISECONDS); }
private void printAllNum() { Thread thread1 = new Thread(new Runnable() { public void run() { try { lock.lock(); printNum(condition1, condition2, (short) 1); } catch (InterruptedException e) { e.printStackTrace(); } finally { condition3.signal(); lock.unlock(); } } }, "线程1"); Thread thread2 = new Thread(new Runnable() { public void run() { try { lock.lock(); printNum(condition2, condition3, (short) 2); } catch (InterruptedException e) { e.printStackTrace(); } finally { condition1.signal(); lock.unlock(); } } }, "线程2"); Thread thread3 = new Thread(new Runnable() { public void run() { try { lock.lock(); printNum(condition3, condition1, (short) 3); } catch (InterruptedException e) { e.printStackTrace(); } finally { condition2.signal(); lock.unlock(); } } }, "线程3"); thread1.start(); thread2.start(); thread3.start(); }
public static void main(String[] args) { ConditionPrintNum printNum = new ConditionPrintNum(); printNum.printAllNum(); } }
|
运行效果如下(无特殊等待时间):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
| 2019-09-22 17:43:14 线程1:1 2019-09-22 17:43:14 线程1:2 2019-09-22 17:43:14 线程1:3 2019-09-22 17:43:14 线程1:4 2019-09-22 17:43:14 线程1:5 2019-09-22 17:43:14 线程2:6 2019-09-22 17:43:14 线程2:7 2019-09-22 17:43:14 线程2:8 2019-09-22 17:43:14 线程2:9 2019-09-22 17:43:14 线程2:10 2019-09-22 17:43:14 线程3:11 2019-09-22 17:43:14 线程3:12 2019-09-22 17:43:14 线程3:13 2019-09-22 17:43:14 线程3:14 2019-09-22 17:43:14 线程3:15 2019-09-22 17:43:14 线程1:16 2019-09-22 17:43:14 线程1:17 2019-09-22 17:43:14 线程1:18 2019-09-22 17:43:14 线程1:19 2019-09-22 17:43:14 线程1:20 2019-09-22 17:43:14 线程2:21 2019-09-22 17:43:14 线程2:22 2019-09-22 17:43:14 线程2:23 2019-09-22 17:43:14 线程2:24 2019-09-22 17:43:14 线程2:25 2019-09-22 17:43:14 线程3:26 2019-09-22 17:43:14 线程3:27 2019-09-22 17:43:14 线程3:28 2019-09-22 17:43:14 线程3:29 2019-09-22 17:43:14 线程3:30 2019-09-22 17:43:14 线程1:31 2019-09-22 17:43:14 线程1:32 2019-09-22 17:43:14 线程1:33 2019-09-22 17:43:14 线程1:34 2019-09-22 17:43:14 线程1:35 2019-09-22 17:43:14 线程2:36 2019-09-22 17:43:14 线程2:37 2019-09-22 17:43:14 线程2:38 2019-09-22 17:43:14 线程2:39 2019-09-22 17:43:14 线程2:40 2019-09-22 17:43:14 线程3:41 2019-09-22 17:43:14 线程3:42 2019-09-22 17:43:14 线程3:43 2019-09-22 17:43:14 线程3:44 2019-09-22 17:43:14 线程3:45 2019-09-22 17:43:14 线程1:46 2019-09-22 17:43:14 线程1:47 2019-09-22 17:43:14 线程1:48 2019-09-22 17:43:14 线程1:49 2019-09-22 17:43:14 线程1:50 2019-09-22 17:43:14 线程2:51 2019-09-22 17:43:14 线程2:52 2019-09-22 17:43:14 线程2:53 2019-09-22 17:43:14 线程2:54 2019-09-22 17:43:14 线程2:55 2019-09-22 17:43:14 线程3:56 2019-09-22 17:43:14 线程3:57 2019-09-22 17:43:14 线程3:58 2019-09-22 17:43:14 线程3:59 2019-09-22 17:43:14 线程3:60 2019-09-22 17:43:14 线程1:61 2019-09-22 17:43:14 线程1:62 2019-09-22 17:43:14 线程1:63 2019-09-22 17:43:14 线程1:64 2019-09-22 17:43:14 线程1:65 2019-09-22 17:43:14 线程2:66 2019-09-22 17:43:14 线程2:67 2019-09-22 17:43:14 线程2:68 2019-09-22 17:43:14 线程2:69 2019-09-22 17:43:14 线程2:70 2019-09-22 17:43:14 线程3:71 2019-09-22 17:43:14 线程3:72 2019-09-22 17:43:14 线程3:73 2019-09-22 17:43:14 线程3:74 2019-09-22 17:43:14 线程3:75
|